001 /* 002 /* 003 * Copyright 2005 Stephen J. McConnell 004 * 005 * Licensed under the Apache License, Version 2.0 (the "License"); 006 * you may not use this file except in compliance with the License. 007 * You may obtain a copy of the License at 008 * 009 * http://www.apache.org/licenses/LICENSE-2.0 010 * 011 * Unless required by applicable law or agreed to in writing, software 012 * distributed under the License is distributed on an "AS IS" BASIS, 013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or 014 * implied. 015 * 016 * See the License for the specific language governing permissions and 017 * limitations under the License. 018 */ 019 020 package net.dpml.library.info; 021 022 import java.io.File; 023 import java.io.IOException; 024 import java.io.FileNotFoundException; 025 import java.net.URI; 026 import java.net.URISyntaxException; 027 import java.util.List; 028 import java.util.ArrayList; 029 import java.util.Properties; 030 031 import net.dpml.library.Feature; 032 import net.dpml.library.TypeBuilder; 033 import net.dpml.library.info.ResourceDirective.Classifier; 034 035 import net.dpml.lang.Category; 036 import net.dpml.lang.Part; 037 import net.dpml.lang.Version; 038 039 import net.dpml.util.DecodingException; 040 import net.dpml.util.DOM3DocumentBuilder; 041 import net.dpml.util.ElementHelper; 042 043 import org.w3c.dom.Element; 044 import org.w3c.dom.Document; 045 import org.w3c.dom.TypeInfo; 046 047 /** 048 * Utility class used for construction of a module model from an XML source. 049 * 050 * @author <a href="http://www.dpml.net">Digital Product Meta Library</a> 051 * @version 1.1.0 052 */ 053 public final class LibraryDecoder extends LibraryConstants 054 { 055 private static final DOM3DocumentBuilder DOCUMENT_BUILDER = new DOM3DocumentBuilder(); 056 057 /** 058 * Construct a library directive from XML source. 059 * @param source the XML source file 060 * @return the library directive 061 * @exception IOException if an IO exception occurs 062 */ 063 public LibraryDirective build( File source ) throws IOException 064 { 065 if( null == source ) 066 { 067 throw new NullPointerException( "source" ); 068 } 069 if( !source.exists() ) 070 { 071 throw new FileNotFoundException( source.toString() ); 072 } 073 if( source.isDirectory() ) 074 { 075 final String error = 076 "File [" 077 + source 078 + "] references a directory."; 079 throw new IllegalArgumentException( error ); 080 } 081 try 082 { 083 final Element root = getRootElement( source ); 084 File base = source.getParentFile(); 085 return buildLibraryDirective( base, root ); 086 } 087 catch( Exception e ) 088 { 089 final String error = 090 "Unexpected error while attempting to load library." 091 + "File: '" + source + "'"; 092 IOException ioe = new IOException( error ); 093 ioe.initCause( e ); 094 throw ioe; 095 } 096 } 097 098 /** 099 * Construct a resource directive from source. 100 * @param uri the source uri 101 * @return the resource directive 102 * @exception IOException if an IO exception occurs 103 */ 104 public ResourceDirective buildResource( URI uri ) throws IOException 105 { 106 try 107 { 108 final Document document = DOCUMENT_BUILDER.parse( uri ); 109 Element root = document.getDocumentElement(); 110 return buildResourceDirectiveFromElement( null, root, null ); 111 } 112 catch( Exception e ) 113 { 114 final String error = 115 "Unexpected error while attempting to load module." 116 + "URI: '" + uri + "'"; 117 IOException ioe = new IOException( error ); 118 ioe.initCause( e ); 119 throw ioe; 120 } 121 } 122 123 /** 124 * Resolve the root DOM element of the supplied file. 125 * @param source the source XML file 126 * @return the root element 127 * @exception IOException if an io error occurs 128 */ 129 private Element getRootElement( File source ) throws IOException 130 { 131 File file = source.getCanonicalFile(); 132 final Document document = DOCUMENT_BUILDER.parse( file.toURI() ); 133 return document.getDocumentElement(); 134 } 135 136 /** 137 * Build a library directive using an XML element. 138 * @param base the base directory 139 * @param element the module element 140 * @return the library directive 141 * @exception Exception if an exception occurs 142 */ 143 private LibraryDirective buildLibraryDirective( File base, Element element ) throws Exception 144 { 145 final String elementName = element.getTagName(); 146 if( !LIBRARY_ELEMENT_NAME.equals( elementName ) ) 147 { 148 final String error = 149 "Element is not a library."; 150 throw new IllegalArgumentException( error ); 151 } 152 153 // get type descriptors, modules and properties 154 155 Properties properties = null; 156 ImportDirective[] imports = new ImportDirective[0]; 157 List list = new ArrayList(); 158 Element[] children = ElementHelper.getChildren( element ); 159 for( int i=0; i<children.length; i++ ) 160 { 161 Element child = children[i]; 162 final String tag = child.getTagName(); 163 if( PROPERTIES_ELEMENT_NAME.equals( tag ) ) 164 { 165 properties = buildProperties( child ); 166 } 167 else if( IMPORTS_ELEMENT_NAME.equals( tag ) ) 168 { 169 imports = buildImportDirectives( child ); 170 } 171 else 172 { 173 ResourceDirective resource = buildResourceDirectiveFromElement( base, child, null ); 174 list.add( resource ); 175 } 176 } 177 ResourceDirective[] resources = (ResourceDirective[]) list.toArray( new ResourceDirective[0] ); 178 return new LibraryDirective( imports, resources, properties ); 179 } 180 181 /** 182 * Build a resource using an XML element. 183 * @param base the base directory 184 * @param element the module element 185 * @param offset the imported file directory offset 186 * @throws Exception if an error occurs 187 */ 188 private ResourceDirective buildResourceDirectiveFromElement( 189 File base, Element element, String offset ) throws Exception 190 { 191 final String elementName = element.getTagName(); 192 final String path = ElementHelper.getAttribute( element, "file" ); 193 if( null != path ) 194 { 195 try 196 { 197 File file = new File( base, path ); 198 File source = file.getCanonicalFile(); 199 if( !source.exists() ) 200 { 201 final String error = 202 "Local file does not exist." 203 + "\n base: " + base 204 + "\n path: " + path 205 + "\n source: " + source; 206 throw new DecodingException( element, error ); 207 } 208 else if( source.isDirectory() ) 209 { 210 final String error = 211 "Local file references a directory." 212 + "\n base: " + base 213 + "\n path: " + path 214 + "\n source: " + source; 215 throw new DecodingException( element, error ); 216 } 217 else 218 { 219 final File parent = source.getParentFile(); 220 final Element root = getRootElement( source ); 221 String basedir = getRelativePath( base, parent ); 222 if( null != offset ) 223 { 224 basedir = offset + "/" + basedir; 225 } 226 return buildResourceDirectiveFromElement( base, root, basedir ); 227 } 228 } 229 catch( DecodingException e ) 230 { 231 throw e; 232 } 233 catch( Throwable e ) 234 { 235 final String error = 236 "Internal error while attempting to resolve a import directive."; 237 throw new DecodingException( element, error, e ); 238 } 239 } 240 else 241 { 242 return buildResourceDirective( base, element, offset ); 243 } 244 } 245 246 private ResourceDirective buildResourceDirective( File base, Element element ) throws Exception 247 { 248 return buildResourceDirective( base, element, null ); 249 } 250 251 private ResourceDirective buildResourceDirective( File base, Element element, String path ) throws Exception 252 { 253 Classifier classifier = null; 254 final String tag = element.getTagName(); 255 if( RESOURCE_ELEMENT_NAME.equals( tag ) || PROJECT_ELEMENT_NAME.equals( tag ) 256 || MODULE_ELEMENT_NAME.equals( tag ) ) 257 { 258 final String name = ElementHelper.getAttribute( element, "name" ); 259 final String version = ElementHelper.getAttribute( element, "version" ); 260 String basedir = ElementHelper.getAttribute( element, "basedir" ); 261 if( path != null ) 262 { 263 if( basedir == null ) 264 { 265 basedir = path; 266 } 267 else 268 { 269 basedir = path + "/" + basedir; 270 } 271 } 272 273 if( PROJECT_ELEMENT_NAME.equals( tag ) ) 274 { 275 classifier = Classifier.LOCAL; 276 if( null == basedir ) 277 { 278 basedir = "."; 279 } 280 } 281 else if( MODULE_ELEMENT_NAME.equals( tag ) ) 282 { 283 if( null != basedir ) 284 { 285 classifier = Classifier.LOCAL; 286 } 287 else 288 { 289 classifier = Classifier.EXTERNAL; 290 } 291 } 292 else 293 { 294 classifier = Classifier.EXTERNAL; 295 } 296 297 final InfoDirective info = 298 buildInfoDirective( 299 ElementHelper.getChild( element, "info" ) ); 300 301 final DataDirective[] data = 302 buildDataTypes( 303 ElementHelper.getChild( element, "types" ) ); 304 305 final DependencyDirective[] dependencies = 306 buildDependencyDirectives( 307 ElementHelper.getChild( element, "dependencies" ) ); 308 309 final FilterDirective[] filters = 310 buildFilterDirectives( 311 ElementHelper.getChild( element, "filters" ) ); 312 313 final Properties properties = 314 buildProperties( 315 ElementHelper.getChild( element, "properties" ) ); 316 317 if( MODULE_ELEMENT_NAME.equals( tag ) ) 318 { 319 File anchor = getAnchorDirectory( base, basedir ); 320 ArrayList list = new ArrayList(); 321 Element[] children = ElementHelper.getChildren( element ); 322 for( int i=0; i<children.length; i++ ) 323 { 324 Element child = children[i]; 325 final String t = child.getTagName(); 326 327 if( RESOURCE_ELEMENT_NAME.equals( t ) 328 || PROJECT_ELEMENT_NAME.equals( t ) 329 || MODULE_ELEMENT_NAME.equals( t ) ) 330 { 331 ResourceDirective directive = 332 buildResourceDirectiveFromElement( anchor, child, null ); 333 list.add( directive ); 334 } 335 } 336 337 ResourceDirective[] resources = 338 (ResourceDirective[]) list.toArray( new ResourceDirective[0] ); 339 return ModuleDirective.createModuleDirective( 340 name, version, classifier, basedir, info, data, dependencies, 341 properties, filters, resources ); 342 } 343 else 344 { 345 return ResourceDirective.createResourceDirective( 346 name, version, classifier, basedir, info, data, dependencies, 347 properties, filters ); 348 } 349 } 350 else 351 { 352 final String error = 353 "Invalid element name [" 354 + tag 355 + "]."; 356 throw new DecodingException( element, error ); 357 } 358 } 359 360 private File getAnchorDirectory( File base, String path ) throws IOException 361 { 362 if( null == base ) 363 { 364 return null; 365 } 366 if( !base.exists() ) 367 { 368 final String error = 369 "Base directory [" + base + "] does not exist."; 370 throw new IllegalArgumentException( error ); 371 } 372 if( null == path ) 373 { 374 return base; 375 } 376 else 377 { 378 return new File( base, path ).getCanonicalFile(); 379 } 380 } 381 382 private String getRelativePath( File base, File dir ) throws IOException 383 { 384 String baseSpec = base.getCanonicalPath(); 385 String dirSpec = dir.getCanonicalPath(); 386 if( dirSpec.equals( baseSpec ) ) 387 { 388 return "."; 389 } 390 if( dirSpec.startsWith( baseSpec ) ) 391 { 392 return dirSpec.substring( baseSpec.length() + 1 ); 393 } 394 else 395 { 396 final String error = 397 "Supplied dir [" + dirSpec + "] is not with base [" + baseSpec + "]."; 398 throw new IllegalArgumentException( error ); 399 } 400 } 401 402 /** 403 * Build an array of include directives contained within the supplied enclosing element. 404 * @param element the enclosing element 405 * @return the array of includes 406 */ 407 private ImportDirective[] buildImportDirectives( Element element ) throws DecodingException 408 { 409 Element[] children = ElementHelper.getChildren( element ); 410 ImportDirective[] includes = new ImportDirective[ children.length ]; 411 for( int i=0; i<children.length; i++ ) 412 { 413 Element child = children[i]; 414 includes[i] = buildImportDirective( child ); 415 } 416 return includes; 417 } 418 419 private ImportDirective buildImportDirective( Element element ) throws DecodingException 420 { 421 final String tag = element.getTagName(); 422 final Properties properties = buildProperties( element ); 423 if( IMPORT_ELEMENT_NAME.equals( tag ) ) 424 { 425 try 426 { 427 if( element.hasAttribute( "file" ) ) 428 { 429 final String value = ElementHelper.getAttribute( element, "file", null ); 430 return new ImportDirective( ImportDirective.FILE, value, properties ); 431 } 432 else if( element.hasAttribute( "uri" ) ) 433 { 434 final String value = ElementHelper.getAttribute( element, "uri", null ); 435 return new ImportDirective( ImportDirective.URI, value, properties ); 436 } 437 else 438 { 439 final String error = 440 "Import element does not declare a 'file' or 'uri' attribute.\n" 441 + element.toString(); 442 throw new IllegalArgumentException( error ); 443 } 444 } 445 catch( Throwable e ) 446 { 447 final String error = 448 "Internal error while attempting to resolve an import directive."; 449 throw new DecodingException( element, error, e ); 450 } 451 } 452 else 453 { 454 final String error = 455 "Invalid include element name [" 456 + tag 457 + "]."; 458 throw new IllegalArgumentException( error ); 459 } 460 } 461 462 private DependencyDirective[] buildDependencyDirectives( Element element ) throws DecodingException 463 { 464 if( null == element ) 465 { 466 return new DependencyDirective[0]; 467 } 468 else 469 { 470 final String tag = element.getTagName(); 471 if( DEPENDENCIES_ELEMENT_NAME.equals( tag ) ) 472 { 473 Element[] children = ElementHelper.getChildren( element ); 474 DependencyDirective[] dependencies = new DependencyDirective[ children.length ]; 475 for( int i=0; i<children.length; i++ ) 476 { 477 Element child = children[i]; 478 dependencies[i] = buildDependencyDirective( child ); 479 } 480 return dependencies; 481 } 482 else 483 { 484 final String error = 485 "Invalid dependency element name [" 486 + tag 487 + "]."; 488 throw new IllegalArgumentException( error ); 489 } 490 } 491 } 492 493 private DependencyDirective buildDependencyDirective( Element element ) throws DecodingException 494 { 495 String name = element.getTagName(); 496 Scope scope = Scope.parse( name ); 497 if( Scope.BUILD.equals( scope ) ) 498 { 499 IncludeDirective[] includes = buildIncludeDirectives( element, false ); 500 return new DependencyDirective( Scope.BUILD, includes ); 501 } 502 else if( Scope.RUNTIME.equals( scope ) ) 503 { 504 IncludeDirective[] includes = buildIncludeDirectives( element, true ); 505 return new DependencyDirective( Scope.RUNTIME, includes ); 506 } 507 else 508 { 509 IncludeDirective[] includes = buildIncludeDirectives( element, false ); 510 return new DependencyDirective( Scope.TEST, includes ); 511 } 512 } 513 514 /** 515 * Build an array of include directives contained within the supplied enclosing element. 516 * @param element the enclosing element 517 * @param flag if category processing is required 518 * @return the array of includes 519 */ 520 private IncludeDirective[] buildIncludeDirectives( Element element, boolean flag ) throws DecodingException 521 { 522 Element[] children = ElementHelper.getChildren( element ); 523 IncludeDirective[] includes = new IncludeDirective[ children.length ]; 524 for( int i=0; i<children.length; i++ ) 525 { 526 Element child = children[i]; 527 includes[i] = buildIncludeDirective( child, flag ); 528 } 529 return includes; 530 } 531 532 private IncludeDirective buildIncludeDirective( Element element, boolean flag ) throws DecodingException 533 { 534 final String tag = element.getTagName(); 535 final Properties properties = buildProperties( element ); 536 if( INCLUDE_ELEMENT_NAME.equals( tag ) ) 537 { 538 Category category = buildCategory( element, flag ); 539 if( element.hasAttribute( "key" ) ) 540 { 541 final String value = ElementHelper.getAttribute( element, "key", null ); 542 return new IncludeDirective( IncludeDirective.KEY, category, value, properties ); 543 } 544 else if( element.hasAttribute( "ref" ) ) 545 { 546 final String value = ElementHelper.getAttribute( element, "ref", null ); 547 return new IncludeDirective( IncludeDirective.REF, category, value, properties ); 548 } 549 else if( element.hasAttribute( "uri" ) ) 550 { 551 final String value = ElementHelper.getAttribute( element, "uri", null ); 552 return new IncludeDirective( IncludeDirective.URI, category, value, properties ); 553 } 554 else 555 { 556 final String error = 557 "Include directive does not declare a 'uri', 'key' or 'ref' attribute.\n" 558 + element.toString(); 559 throw new IllegalArgumentException( error ); 560 } 561 } 562 else 563 { 564 final String error = 565 "Invalid include element name [" 566 + tag 567 + "]."; 568 throw new DecodingException( element, error ); 569 } 570 } 571 572 private Category buildCategory( Element element, boolean flag ) 573 { 574 if( !flag ) 575 { 576 return Category.UNDEFINED; 577 } 578 else 579 { 580 final String value = ElementHelper.getAttribute( element, "tag", "private" ); 581 return Category.parse( value ); 582 } 583 } 584 585 private InfoDirective buildInfoDirective( Element element ) 586 { 587 if( null == element ) 588 { 589 return null; 590 } 591 else 592 { 593 String title = ElementHelper.getAttribute( element, "title" ); 594 Element child = ElementHelper.getChild( element, "description" ); 595 if( null == child ) 596 { 597 return new InfoDirective( title, null ); 598 } 599 else 600 { 601 String value = ElementHelper.getValue( child ); 602 String description = trim( value ); 603 return new InfoDirective( title, description ); 604 } 605 } 606 } 607 608 private String trim( String value ) 609 { 610 if( null == value ) 611 { 612 return null; 613 } 614 String trimmed = value.trim(); 615 if( trimmed.startsWith( "\n" ) ) 616 { 617 return trim( trimmed.substring( 1 ) ); 618 } 619 else if( trimmed.endsWith( "\n" ) ) 620 { 621 return trim( trimmed.substring( 0, trimmed.length() - 1 ) ); 622 } 623 else 624 { 625 return trimmed; 626 } 627 } 628 629 private FilterDirective[] buildFilterDirectives( Element element ) throws Exception 630 { 631 if( null == element ) 632 { 633 return new FilterDirective[0]; 634 } 635 else 636 { 637 Element[] children = ElementHelper.getChildren( element ); 638 FilterDirective[] filters = new FilterDirective[ children.length ]; 639 for( int i=0; i<children.length; i++ ) 640 { 641 Element child = children[i]; 642 String token = ElementHelper.getAttribute( child, "token" ); 643 if( "filter".equals( child.getTagName() ) ) 644 { 645 String value = ElementHelper.getAttribute( child, "value" ); 646 filters[i] = new SimpleFilterDirective( token, value ); 647 } 648 else if( "feature".equals( child.getTagName() ) ) 649 { 650 String id = ElementHelper.getAttribute( child, "id" ); 651 Feature feature = Feature.parse( id ); 652 String ref = ElementHelper.getAttribute( child, "ref" ); 653 String type = ElementHelper.getAttribute( child, "type" ); 654 boolean alias = ElementHelper.getBooleanAttribute( child, "alias" ); 655 filters[i] = new FeatureFilterDirective( token, ref, feature, type, alias ); 656 } 657 else 658 { 659 final String error = 660 "Element name not recognized [" 661 + child.getTagName() 662 + "] (expecting 'filter')."; 663 throw new DecodingException( element, error ); 664 } 665 } 666 return filters; 667 } 668 } 669 670 private DataDirective[] buildDataTypes( Element element ) throws Exception 671 { 672 if( null == element ) 673 { 674 return new DataDirective[0]; 675 } 676 Element[] children = ElementHelper.getChildren( element ); 677 DataDirective[] data = new DataDirective[ children.length ]; 678 for( int i=0; i<children.length; i++ ) 679 { 680 Element child = children[i]; 681 data[i] = buildDataType( child ); 682 } 683 return data; 684 } 685 686 private DataDirective buildDataType( Element element ) throws Exception 687 { 688 TypeInfo info = element.getSchemaTypeInfo(); 689 String namespace = info.getTypeNamespace(); 690 String typeName = info.getTypeName(); 691 692 if( null == namespace ) 693 { 694 throw new NullPointerException( "namespace" ); 695 } 696 697 if( MODULE_XSD_URI.equals( namespace ) ) 698 { 699 // it's a generic type declaration 700 701 final String id = getID( element ); 702 final Properties properties = getProperties( element ); 703 final Version version = getVersion( element ); 704 return new TypeDirective( id, version, properties ); 705 } 706 else if( info.isDerivedFrom( MODULE_XSD_URI, "AbstractType", TypeInfo.DERIVATION_EXTENSION ) ) 707 { 708 if( 709 PART_XSD_URI.equals( namespace ) 710 || info.isDerivedFrom( PART_XSD_URI, "StrategyType", TypeInfo.DERIVATION_EXTENSION ) ) 711 { 712 final Version version = getVersion( element ); 713 return new TypeDirective( element, "part", version ); 714 } 715 else 716 { 717 // id attribute is required 718 719 final String id = getID( element ); 720 final Version version = getVersion( element ); 721 return new TypeDirective( element, id, version ); 722 } 723 } 724 else 725 { 726 final String error = 727 "Unsupported element encountered during type directive production." 728 + "\nNamespace: " 729 + namespace 730 + "\nType: " 731 + info.getTypeName() 732 + "\nName: " 733 + element.getTagName(); 734 throw new DecodingException( element, error ); 735 } 736 } 737 738 private Version getVersion( Element element ) 739 { 740 final String version = ElementHelper.getAttribute( element, "version" ); 741 if( null != version ) 742 { 743 return Version.parse( version ); 744 } 745 else if( getAliasFlag( element ) ) 746 { 747 return Version.NULL_VERSION; 748 } 749 else 750 { 751 return null; 752 } 753 } 754 755 private Properties buildProperties( Element element ) 756 { 757 if( null == element ) 758 { 759 return null; 760 } 761 Properties properties = new Properties(); 762 Element[] children = ElementHelper.getChildren( element ); 763 for( int i=0; i<children.length; i++ ) 764 { 765 Element child = children[i]; 766 String tag = child.getTagName(); 767 if( "property".equals( tag ) ) 768 { 769 String key = ElementHelper.getAttribute( child, "name", null ); 770 if( null == key ) 771 { 772 final String error = 773 "Property declaration does not contain a 'name' attribute."; 774 throw new IllegalArgumentException( error ); 775 } 776 else 777 { 778 String value = ElementHelper.getAttribute( child, "value", null ); 779 properties.setProperty( key, value ); 780 } 781 } 782 } 783 return properties; 784 } 785 786 private URI getURIFromSpec( String spec ) 787 { 788 if( null == spec ) 789 { 790 return null; 791 } 792 else 793 { 794 try 795 { 796 return new URI( spec ); 797 } 798 catch( URISyntaxException e ) 799 { 800 final String error = 801 "Type descriptor uri [" 802 + spec 803 + "] could not be converted to a URI value."; 804 throw new IllegalArgumentException( error ); 805 } 806 } 807 } 808 809 private TypeBuilder loadTypeBuilder( URI uri ) throws Exception 810 { 811 ClassLoader classloader = TypeBuilder.class.getClassLoader(); 812 Object[] args = new Object[0]; 813 Part part = Part.load( uri ); 814 Object handler = part.instantiate( args ); 815 if( handler instanceof TypeBuilder ) 816 { 817 return (TypeBuilder) handler; 818 } 819 else 820 { 821 final String error = 822 "Plugin [" 823 + uri 824 + "] does not implement the " 825 + TypeBuilder.class.getName() 826 + " interface."; 827 throw new IllegalArgumentException( error ); 828 } 829 } 830 831 /** 832 * Return the id attribute of the supplied element. 833 * @param element the DOM element 834 * @return the id value 835 * @exception DecodingException if an error occurs during element evaluation 836 */ 837 protected String getID( Element element ) throws DecodingException 838 { 839 final String id = ElementHelper.getAttribute( element, "id" ); 840 if( null == id ) 841 { 842 final String error = 843 "Missing type 'id'."; 844 throw new DecodingException( element, error ); 845 } 846 else 847 { 848 return id; 849 } 850 } 851 852 /** 853 * Return the alias attribute of the supplied element. 854 * @param element the DOM element 855 * @return the alias production flag value 856 */ 857 protected boolean getAliasFlag( Element element ) 858 { 859 return ElementHelper.getBooleanAttribute( element, "alias", false ); 860 } 861 862 /** 863 * Return properties declared within the supplied element as immediate 864 * child <property> elements. 865 * @param element the enclosing DOM element 866 * @return the resolved properties instance 867 */ 868 protected Properties getProperties( Element element ) 869 { 870 Properties properties = new Properties(); 871 Element[] children = ElementHelper.getChildren( element ); 872 for( int i=0; i<children.length; i++ ) 873 { 874 Element child = children[i]; 875 String tag = child.getTagName(); 876 if( "property".equals( tag ) ) 877 { 878 String key = ElementHelper.getAttribute( child, "name", null ); 879 if( null == key ) 880 { 881 final String error = 882 "Property declaration does not contain a 'name' attribute."; 883 throw new IllegalArgumentException( error ); 884 } 885 else 886 { 887 String value = ElementHelper.getAttribute( child, "value", null ); 888 properties.setProperty( key, value ); 889 } 890 } 891 } 892 return properties; 893 } 894 }